home *** CD-ROM | disk | FTP | other *** search
/ CU Amiga Super CD-ROM 16 / CU Amiga Magazine's Super CD-ROM 16 (1997-10-16)(EMAP Images)(GB)[!][issue 1997-11].iso / CUCD / Graphics / Ghostscript / source / gp_mswin.c < prev    next >
C/C++ Source or Header  |  1997-08-05  |  22KB  |  853 lines

  1. /* Copyright (C) 1992, 1995, 1996, 1997 Aladdin Enterprises.  All rights reserved.
  2.   
  3.   This file is part of Aladdin Ghostscript.
  4.   
  5.   Aladdin Ghostscript is distributed with NO WARRANTY OF ANY KIND.  No author
  6.   or distributor accepts any responsibility for the consequences of using it,
  7.   or for whether it serves any particular purpose or works at all, unless he
  8.   or she says so in writing.  Refer to the Aladdin Ghostscript Free Public
  9.   License (the "License") for full details.
  10.   
  11.   Every copy of Aladdin Ghostscript must include a copy of the License,
  12.   normally in a plain ASCII text file named PUBLIC.  The License grants you
  13.   the right to copy, modify and redistribute Aladdin Ghostscript, but only
  14.   under certain conditions described in the License.  Among other things, the
  15.   License requires that the copyright notice and this notice be preserved on
  16.   all copies.
  17. */
  18.  
  19. /* gp_mswin.c */
  20. /*
  21.  * Microsoft Windows 3.n platform support for Ghostscript.
  22.  * Original version by Russell Lang and Maurice Castro with help from
  23.  * Programming Windows, 2nd Ed., Charles Petzold, Microsoft Press;
  24.  * initially created from gp_dosfb.c and gp_itbc.c 5th June 1992.
  25.  */
  26.  
  27. /* Modified for Win32 & Microsoft C/C++ 8.0 32-Bit, 26.Okt.1994 */
  28. /* by Friedrich Nowak                                           */
  29.  
  30. /* Original EXE and GSview specific code removed */
  31. /* DLL version must now be used under MS-Windows */
  32. /* Russell Lang 16 March 1996 */
  33.  
  34. #include "stdio_.h"
  35. #include "string_.h"
  36. #include "memory_.h"
  37. #include <stdlib.h>
  38. #include <stdarg.h>
  39. #include "ctype_.h"
  40. #include "dos_.h"
  41. #include <io.h>
  42. #include "malloc_.h"
  43. #include <fcntl.h>
  44. #include <signal.h>
  45. #include "gx.h"
  46. #include "gp.h"
  47. #include "gpcheck.h"
  48. #include "gserrors.h"
  49. #include "gsexit.h"
  50.  
  51. #include "windows_.h"
  52. #include <shellapi.h>
  53. #ifdef __WIN32__
  54. #include <winspool.h>
  55. #endif
  56. #include "gp_mswin.h"
  57. #include "gsdll.h"
  58. /* use longjmp instead of exit when using DLL */
  59. #include <setjmp.h>
  60. extern jmp_buf gsdll_env;
  61.  
  62. /* Library routines not declared in a standard header */
  63. extern char *getenv(P1(const char *));
  64.  
  65. /* ------ from gnuplot winmain.c plus new stuff ------ */
  66.  
  67. /* limits */
  68. #define MAXSTR 255
  69.  
  70. /* public handles */
  71. HINSTANCE phInstance;
  72. HWND hwndtext = HWND_DESKTOP;    /* would be better to be a real window */
  73.  
  74. const LPSTR szAppName = "Ghostscript";
  75. BOOL is_win32s = FALSE;
  76. char FAR win_prntmp[MAXSTR];    /* filename of PRN temporary file */
  77. private int is_printer(const char *name);
  78. int win_init = 0;        /* flag to know if gp_exit has been called */
  79. int win_exit_status;
  80.  
  81. BOOL CALLBACK _export AbortProc(HDC, int);
  82.  
  83. #ifdef __WIN32__
  84. /* DLL entry point for Borland C++ */
  85. BOOL WINAPI _export
  86. DllEntryPoint(HINSTANCE hInst, DWORD fdwReason, LPVOID lpReserved)
  87. {
  88.     /* Win32s: HIWORD bit 15 is 1 and bit 14 is 0 */
  89.     /* Win95:  HIWORD bit 15 is 1 and bit 14 is 1 */
  90.     /* WinNT:  HIWORD bit 15 is 0 and bit 14 is 0 */
  91.     /* WinNT Shell Update Release is WinNT && LOBYTE(LOWORD) >= 4 */
  92.     DWORD version = GetVersion();
  93.     if ( ((HIWORD(version) & 0x8000) != 0) && ((HIWORD(version) & 0x4000) == 0) )
  94.         is_win32s = TRUE;
  95.  
  96.     phInstance = hInst;
  97.     return TRUE;
  98. }
  99.  
  100. /* DLL entry point for Microsoft Visual C++ */
  101. BOOL WINAPI _export
  102. DllMain(HINSTANCE hInst, DWORD fdwReason, LPVOID lpReserved)
  103. {
  104.     DllEntryPoint(hInst, fdwReason, lpReserved);
  105. }
  106.  
  107.  
  108. #else
  109. int WINAPI _export
  110. LibMain(HINSTANCE hInstance, WORD wDataSeg, WORD wHeapSize, LPSTR lpszCmdLine)
  111. {
  112.     phInstance = hInstance;
  113.     return 1;
  114. }
  115.  
  116. int WINAPI _export
  117. WEP(int nParam)
  118. {
  119.     return 1;
  120. }
  121. #endif
  122.  
  123.  
  124. BOOL CALLBACK _export
  125. AbortProc(HDC hdcPrn, int code)
  126. {
  127.     process_interrupts();
  128.     if (code == SP_OUTOFDISK)
  129.     return (FALSE);    /* cancel job */
  130.     return(TRUE);
  131. }
  132.   
  133. /* ------ Process message loop ------ */
  134. /*
  135.  * Check messages and interrupts; return true if interrupted.
  136.  * This is called frequently - it must be quick!
  137.  */
  138. int
  139. gp_check_interrupts(void)
  140. {
  141.     return (*pgsdll_callback)(GSDLL_POLL, NULL, 0);
  142. }
  143.  
  144. /* ====== Generic platform procedures ====== */
  145.  
  146. /* ------ Initialization/termination (from gp_itbc.c) ------ */
  147.  
  148. /* Do platform-dependent initialization. */
  149. void
  150. gp_init(void)
  151. {
  152.     win_init = 1;
  153. }
  154.  
  155. /* Do platform-dependent cleanup. */
  156. void
  157. gp_exit(int exit_status, int code)
  158. {
  159.     win_init = 0;
  160.     win_exit_status = exit_status;
  161. }
  162.  
  163. /* Exit the program. */
  164. void
  165. gp_do_exit(int exit_status)
  166. {
  167.     /* Use longjmp since exit would terminate caller */
  168.     /* setjmp code will check gs_exit_status */
  169.     longjmp(gsdll_env, gs_exit_status);
  170. }
  171.  
  172. /* ------ Printer accessing ------ */
  173.   
  174. /* Forward references */
  175. private int gp_printfile(P2(const char *, const char *));
  176.  
  177. /* Open a connection to a printer.  A null file name means use the */
  178. /* standard printer connected to the machine, if any. */
  179. /* Return NULL if the connection could not be opened. */
  180. FILE *
  181. gp_open_printer(char *fname, int binary_mode)
  182. {
  183.     if (is_printer(fname)) 
  184.     {    FILE *pfile;
  185.         /* Open a scratch file, which we will send to the */
  186.         /* actual printer in gp_close_printer. */
  187.         pfile = gp_open_scratch_file(gp_scratch_file_name_prefix, 
  188.                          win_prntmp, "wb");
  189.         return pfile;
  190.     }
  191.     else
  192.         return fopen(fname, (binary_mode ? "wb" : "w"));
  193. }
  194.  
  195. /* Close the connection to the printer. */
  196. void
  197. gp_close_printer(FILE *pfile, const char *fname)
  198. {
  199.     fclose(pfile);
  200.     if (!is_printer(fname))
  201.         return;        /* a file, not a printer */
  202.  
  203.     gp_printfile(win_prntmp, fname);
  204.     unlink(win_prntmp);
  205. }
  206.  
  207. /* Printer abort procedure and progress/cancel dialog box */
  208. /* Used by Win32 and mswinprn device */
  209.  
  210. HWND hDlgModeless;
  211.  
  212. BOOL CALLBACK _export
  213. PrintAbortProc(HDC hdcPrn, int code)
  214. {
  215.     MSG msg;
  216.     while (hDlgModeless && PeekMessage(&msg, 0, 0, 0, PM_REMOVE)) {
  217.     if (hDlgModeless || !IsDialogMessage(hDlgModeless,&msg)) {
  218.             TranslateMessage(&msg);
  219.             DispatchMessage(&msg);
  220.     }
  221.     }
  222.     return(hDlgModeless!=0);
  223. }
  224.  
  225. /* Modeless dialog box - Cancel printing */
  226. BOOL CALLBACK _export
  227. CancelDlgProc(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
  228. {
  229.     switch(message) {
  230.     case WM_INITDIALOG:
  231.         SetWindowText(hDlg, szAppName);
  232.         return TRUE;
  233.     case WM_COMMAND:
  234.         switch(LOWORD(wParam)) {
  235.         case IDCANCEL:
  236.             DestroyWindow(hDlg);
  237.             hDlgModeless = 0;
  238.             EndDialog(hDlg, 0);
  239.             return TRUE;
  240.         }
  241.     }
  242.     return FALSE;
  243. }
  244.  
  245. #ifndef __WIN32__
  246.  
  247. /* Windows does not provide API's in the SDK for writing directly to a */
  248. /* printer.  Instead you are supposed to use the Windows printer drivers. */
  249. /* Ghostscript has its own printer drivers, so we need to use some API's */
  250. /* that are documented only in the Device Driver Adaptation Guide */
  251. /* that comes with the DDK.  Prototypes taken from DDK <print.h> */
  252. DECLARE_HANDLE(HPJOB);
  253.  
  254. HPJOB    WINAPI OpenJob(LPSTR, LPSTR, HPJOB);
  255. int    WINAPI StartSpoolPage(HPJOB);
  256. int    WINAPI EndSpoolPage(HPJOB);
  257. int    WINAPI WriteSpool(HPJOB, LPSTR, int);
  258. int    WINAPI CloseJob(HPJOB);
  259. int    WINAPI DeleteJob(HPJOB, int);
  260. int    WINAPI WriteDialog(HPJOB, LPSTR, int);
  261. int    WINAPI DeleteSpoolPage(HPJOB);
  262.  
  263. #endif        /* WIN32 */
  264.  
  265. /* Dialog box to select printer port */
  266. BOOL CALLBACK _export
  267. SpoolDlgProc(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
  268. {
  269. LPSTR entry;
  270.     switch(message) {
  271.     case WM_INITDIALOG:
  272.         entry = (LPSTR)lParam;
  273.         while (*entry) {
  274.         SendDlgItemMessage(hDlg, SPOOL_PORT, LB_ADDSTRING, 0, (LPARAM)entry);
  275.         entry += lstrlen(entry)+1;
  276.         }
  277.         SendDlgItemMessage(hDlg, SPOOL_PORT, LB_SETCURSEL, 0, (LPARAM)0);
  278.         return TRUE;
  279.     case WM_COMMAND:
  280.         switch(LOWORD(wParam)) {
  281.         case SPOOL_PORT:
  282. #ifdef __WIN32__
  283.             if (HIWORD(wParam)
  284. #else
  285.             if (HIWORD(lParam)
  286. #endif
  287.                            == LBN_DBLCLK)
  288.             PostMessage(hDlg, WM_COMMAND, IDOK, 0L);
  289.             return FALSE;
  290.         case IDOK:
  291.             EndDialog(hDlg, 1+(int)SendDlgItemMessage(hDlg, SPOOL_PORT, LB_GETCURSEL, 0, 0L));
  292.             return TRUE;
  293.         case IDCANCEL:
  294.             EndDialog(hDlg, 0);
  295.             return TRUE;
  296.         }
  297.     }
  298.     return FALSE;
  299. }
  300.  
  301. /* return TRUE if queue looks like a valid printer name */
  302. int
  303. is_spool(const char *queue)
  304. {
  305. char *prefix = "\\\\spool";  /* 7 characters long */
  306. int i;
  307.     for (i=0; i<7; i++) {
  308.     if (prefix[i] == '\\') {
  309.         if ((*queue != '\\') && (*queue != '/'))
  310.             return FALSE;
  311.     }
  312.     else if (tolower(*queue) != prefix[i])
  313.         return FALSE;
  314.     queue++;
  315.     }
  316.     if (*queue && (*queue != '\\') && (*queue != '/'))
  317.     return FALSE;
  318.     return TRUE;
  319. }
  320.  
  321.  
  322. private int 
  323. is_printer(const char *name)
  324. {
  325. char buf[128];
  326.     /* is printer if no name given */
  327.     if (strlen(name) == 0)
  328.     return TRUE;
  329.  
  330.     /*  is printer if name appears in win.ini [ports] section */
  331.     GetProfileString("ports", name, "XYZ", buf, sizeof(buf));
  332.     if ( strlen(name) == 0 || strcmp(buf,"XYZ"))
  333.     return TRUE;
  334.  
  335.     /* is printer if name prefixed by \\spool\ */
  336.     if ( is_spool(name) )
  337.     return TRUE;
  338.  
  339.     return FALSE;
  340. }
  341.  
  342. #ifdef __WIN32__        /* ******** WIN32 ******** */
  343.  
  344. /******************************************************************/
  345. /* Print File to port or queue */
  346. /* port==NULL means prompt for port or queue with dialog box */
  347.  
  348. /* This is messy because Microsoft changed the spooler interface */
  349. /* between Window 3.1 and Windows 95/NT */
  350. /* and didn't provide the spooler interface in Win32s */
  351.  
  352. /* Win95, WinNT: Use OpenPrinter, WritePrinter etc. */
  353. private int gp_printfile_win32(const char *filename, char *port);
  354.  
  355. /* Win32s: Pass to Win16 spooler via gs16spl.exe */ 
  356. private int gp_printfile_gs16spl(const char *filename, const char *port);
  357.  
  358.  
  359. /* valid values for pmport are:
  360.  *   ""
  361.  *      action: WinNT and Win95 use default queue, Win32s prompts for port
  362.  *   "LPT1:" (or other port that appears in win.ini [ports]
  363.  *      action: start gs16spl.exe to print to the port
  364.  *   "\\spool\printer name"
  365.  *      action: send to printer using WritePrinter (WinNT and Win95).
  366.  *      action: translate to port name using win.ini [Devices]
  367.  *              then use gs16spl.exe (Win32s).
  368.  *   "\\spool"
  369.  *      action: prompt for queue name then send to printer using 
  370.  *              WritePrinter (WinNT and Win95).
  371.  *      action: prompt for port then use gs16spl.exe (Win32s).
  372.  *   
  373. /* Print File */
  374. private int
  375. gp_printfile(const char *filename, const char *pmport)
  376. {
  377.     /* treat WinNT and Win95 differently to Win32s */
  378.         if (!is_win32s) {
  379.         if (strlen(pmport)==0) { /* get default printer */
  380.         char buf[256];
  381.         char *p;
  382.         /* WinNT stores default printer in registry and win.ini */
  383.         /* Win95 stores default printer in win.ini */
  384.             GetProfileString("windows", "device", "", buf, sizeof(buf));
  385.         if ( (p = strchr(buf, ',')) != NULL )
  386.             *p = '\0';
  387.             return gp_printfile_win32(filename, buf);
  388.         }
  389.         else if (is_spool(pmport))  {
  390.          if (strlen(pmport) >= 8)
  391.                 return gp_printfile_win32(filename, (char *)pmport+8);
  392.         else
  393.                 return gp_printfile_win32(filename, (char *)NULL);
  394.         }
  395.         else
  396.             return gp_printfile_gs16spl(filename, pmport);
  397.     }
  398.     else {
  399.         /* Win32s */
  400.         if (is_spool(pmport)) {
  401.         if (strlen(pmport) >= 8) {
  402.             /* extract port name from win.ini */
  403.             char driverbuf[256];
  404.             char *output;
  405.             GetProfileString("Devices", pmport+8, "", driverbuf, sizeof(driverbuf));
  406.             strtok(driverbuf, ",");
  407.             output = strtok(NULL, ",");
  408.             return gp_printfile_gs16spl(filename, output);
  409.         }
  410.         else
  411.             return gp_printfile_gs16spl(filename, (char *)NULL);
  412.         }
  413.         else
  414.         return gp_printfile_gs16spl(filename, pmport);
  415.     }
  416. }
  417.  
  418. #define PRINT_BUF_SIZE 16384u
  419. #define PORT_BUF_SIZE 4096
  420.  
  421. char *
  422. get_queues(void)
  423. {
  424. int i;
  425. DWORD count, needed;
  426. PRINTER_INFO_1 *prinfo;
  427. char *enumbuffer;
  428. char *buffer;
  429. char *p;
  430.     /* enumerate all available printers */
  431.     EnumPrinters(PRINTER_ENUM_CONNECTIONS | PRINTER_ENUM_LOCAL, NULL, 1, NULL, 0, &needed, &count);
  432.     if (needed == 0) {
  433.         /* no printers */
  434.         enumbuffer = malloc(4);
  435.     if (enumbuffer == (char *)NULL)
  436.         return NULL;
  437.         memset(enumbuffer, 0, 4);
  438.         return enumbuffer;
  439.     }
  440.     enumbuffer = malloc(needed);
  441.     if (enumbuffer == (char *)NULL)
  442.     return NULL;
  443.     if (!EnumPrinters(PRINTER_ENUM_CONNECTIONS | PRINTER_ENUM_LOCAL, NULL, 1, (LPBYTE)enumbuffer, needed, &needed, &count)) {
  444.     char buf[256];
  445.     free(enumbuffer);
  446.     sprintf(buf, "EnumPrinters() failed, error code = %d", GetLastError());
  447.     MessageBox((HWND)NULL, buf, szAppName, MB_OK | MB_ICONSTOP);
  448.     return NULL;
  449.     }
  450.     prinfo = (PRINTER_INFO_1 *)enumbuffer;
  451.     if ((buffer = malloc(PORT_BUF_SIZE)) == (char *)NULL) {
  452.     free(enumbuffer);
  453.     return NULL;
  454.     }
  455.     /* copy printer names to single buffer */
  456.     p = buffer;
  457.     for (i=0; i<count; i++) {
  458.     if (strlen(prinfo[i].pName) + 1 < (PORT_BUF_SIZE- (p-buffer))) {
  459.         strcpy(p, prinfo[i].pName);
  460.         p += strlen(p) + 1;
  461.     }
  462.     }
  463.     *p = '\0';    /* double null at end */
  464.     free(enumbuffer);
  465.     return buffer;
  466. }
  467.  
  468.  
  469. char *
  470. get_ports(void)
  471. {
  472. char *buffer;
  473. #ifdef __WIN32__
  474.     if (!is_win32s)
  475.         return get_queues();
  476. #endif
  477.  
  478.     if ((buffer = malloc(PORT_BUF_SIZE)) == (char *)NULL)
  479.         return NULL;
  480.     GetProfileString("ports", NULL, "", buffer, PORT_BUF_SIZE);
  481.     return buffer;
  482. }
  483.  
  484. /* return TRUE if queuename available */
  485. /* return FALSE if cancelled or error */
  486. /* if queue non-NULL, use as suggested queue */
  487. BOOL
  488. get_queuename(char *portname, const char *queue)
  489. {
  490. char *buffer;
  491. char *p;
  492. int i, iport;
  493.  
  494.     buffer = get_queues();
  495.     if (buffer == NULL)
  496.     return FALSE;
  497.     if ( (queue == (char *)NULL) || (strlen(queue)==0) ) {
  498.     /* select a queue */
  499.     iport = DialogBoxParam(phInstance, "QueueDlgBox", (HWND)NULL, SpoolDlgProc, (LPARAM)buffer);
  500.     if (!iport) {
  501.         free(buffer);
  502.         return FALSE;
  503.     }
  504.     p = buffer;
  505.     for (i=1; i<iport && strlen(p)!=0; i++)
  506.         p += lstrlen(p)+1;
  507.     /* prepend \\spool\ which is used by Ghostscript to distinguish */
  508.     /* real files from queues */
  509.     strcpy(portname, "\\\\spool\\");
  510.     strcat(portname, p);
  511.     }
  512.     else {
  513.     strcpy(portname, "\\\\spool\\");
  514.     strcat(portname, queue);
  515.     }
  516.  
  517.     free(buffer);
  518.     return TRUE;
  519. }
  520.  
  521. /* return TRUE if portname available */
  522. /* return FALSE if cancelled or error */
  523. /* if port non-NULL, use as suggested port */
  524. BOOL
  525. get_portname(char *portname, const char *port)
  526. {
  527. char *buffer;
  528. char *p;
  529. int i, iport;
  530. char filename[MAXSTR];
  531.     buffer = get_ports();
  532.     if (buffer == NULL)
  533.         return FALSE;
  534.     if ( (port == (char *)NULL) || (strlen(port)==0) ) {
  535.         if (buffer == (char *)NULL)
  536.         return FALSE;
  537.         /* select a port */
  538.         iport = DialogBoxParam(phInstance, "SpoolDlgBox", (HWND)NULL, SpoolDlgProc, (LPARAM)buffer);
  539.         if (!iport) {
  540.             free(buffer);
  541.             return FALSE;
  542.         }
  543.         p = buffer;
  544.         for (i=1; i<iport && strlen(p)!=0; i++)
  545.             p += lstrlen(p)+1;
  546.         strcpy(portname, p);
  547.     }
  548.     else
  549.         strcpy(portname, port);
  550.  
  551.     if (strlen(portname) == 0)
  552.         return FALSE;
  553.     if (strcmp(portname,"FILE:") == 0) {
  554.         OPENFILENAME ofn;
  555.         filename[0] = '\0';
  556.         memset(&ofn, 0, sizeof(OPENFILENAME));
  557.         ofn.lStructSize = sizeof(OPENFILENAME);
  558.         ofn.hwndOwner = (HWND)NULL;
  559.         ofn.lpstrFile = filename;
  560.         ofn.nMaxFile = sizeof(filename);
  561.         ofn.Flags = OFN_PATHMUSTEXIST;
  562.         if (!GetSaveFileName(&ofn)) {
  563.             free(buffer);
  564.             return FALSE;
  565.         }
  566.         strcpy(portname, filename);
  567.     }
  568.     free(buffer);
  569.     return TRUE;
  570. }
  571.  
  572.  
  573. /* True Win32 method, using OpenPrinter, WritePrinter etc. */
  574. private int 
  575. gp_printfile_win32(const char *filename, char *port)
  576. {
  577. DWORD count;
  578. char *buffer;
  579. char portname[MAXSTR];
  580. FILE *f;
  581. HANDLE printer;
  582. DOC_INFO_1 di;
  583. DWORD written;
  584.  
  585.     if (!get_queuename(portname, port))
  586.     return FALSE;
  587.     port = portname + 8;    /* skip over \\spool\ */
  588.  
  589.     if ((buffer = malloc(PRINT_BUF_SIZE)) == (char *)NULL)
  590.         return FALSE;
  591.     
  592.     if ((f = fopen(filename, "rb")) == (FILE *)NULL) {
  593.     free(buffer);
  594.     return FALSE;
  595.     }
  596.  
  597.     /* open a printer */
  598.     if (!OpenPrinter(port, &printer, NULL)) {
  599.     char buf[256];
  600.     sprintf(buf, "OpenPrinter() failed for \042%s\042, error code = %d", port, GetLastError());
  601.     MessageBox((HWND)NULL, buf, szAppName, MB_OK | MB_ICONSTOP);
  602.     free(buffer);
  603.     return FALSE;
  604.     }
  605.     /* from here until ClosePrinter, should AbortPrinter on error */
  606.  
  607.     di.pDocName = szAppName;
  608.     di.pOutputFile = NULL;
  609.     di.pDatatype = "RAW";  /* for available types see EnumPrintProcessorDatatypes */
  610.     if (!StartDocPrinter(printer, 1, (LPBYTE)&di)) {
  611.     char buf[256];
  612.     sprintf(buf, "StartDocPrinter() failed, error code = %d", GetLastError());
  613.     MessageBox((HWND)NULL, buf, szAppName, MB_OK | MB_ICONSTOP);
  614.     AbortPrinter(printer);
  615.     free(buffer);
  616.     return FALSE;
  617.     }
  618.    
  619.     /* copy file to printer */
  620.     while ((count = fread(buffer, 1, PRINT_BUF_SIZE, f)) != 0 ) {
  621.     if (!WritePrinter(printer, (LPVOID)buffer, count, &written)) {
  622.         free(buffer);
  623.         fclose(f);
  624.         AbortPrinter(printer);
  625.         return FALSE;
  626.     }
  627.     }
  628.     fclose(f);
  629.     free(buffer);
  630.  
  631.     if (!EndDocPrinter(printer)) {
  632.     char buf[256];
  633.     sprintf(buf, "EndDocPrinter() failed, error code = %d", GetLastError());
  634.     MessageBox((HWND)NULL, buf, szAppName, MB_OK | MB_ICONSTOP);
  635.     AbortPrinter(printer);
  636.     return FALSE;
  637.     }
  638.  
  639.     if (!ClosePrinter(printer)) {
  640.     char buf[256];
  641.     sprintf(buf, "ClosePrinter() failed, error code = %d", GetLastError());
  642.     MessageBox((HWND)NULL, buf, szAppName, MB_OK | MB_ICONSTOP);
  643.     return FALSE;
  644.     }
  645.     return TRUE;
  646. }
  647.  
  648.  
  649. /* Start a 16-bit application gs16spl.exe to print a file */
  650. /* Intended for Win32s where 16-bit spooler functions are not available */
  651. /* and Win32 spooler functions are not implemented. */
  652. int
  653. gp_printfile_gs16spl(const char *filename, const char *port)
  654. {
  655. /* Get printer port list from win.ini */
  656. char portname[MAXSTR];
  657. HINSTANCE hinst;
  658. char command[MAXSTR];
  659. char *p;
  660. HWND hwndspl;
  661.  
  662.     if (!get_portname(portname, port))
  663.         return FALSE;
  664.  
  665.     /* get path to EXE - same as DLL */
  666.     GetModuleFileName(phInstance, command, sizeof(command));
  667.     if ((p = strrchr(command,'\\')) != (char *)NULL)
  668.         p++;
  669.     else
  670.         p = command;
  671.     *p = '\0';
  672.     sprintf(command+strlen(command), "gs16spl.exe %s %s", 
  673.         portname, filename);
  674.  
  675.     hinst = (HINSTANCE)WinExec(command, SW_SHOWNORMAL);
  676.     if (hinst < (HINSTANCE)HINSTANCE_ERROR)
  677.     { char buf[MAXSTR];
  678.         sprintf(buf, "Can't run: %s", command);
  679.         MessageBox((HWND)NULL, buf, szAppName, MB_OK | MB_ICONSTOP);
  680.         return FALSE;
  681.     }
  682.  
  683.     hwndspl = FindWindow(NULL, "GS Win32s/Win16 spooler");
  684.  
  685.     while (IsWindow(hwndspl)) {
  686.         gp_check_interrupts();
  687.     }
  688.  
  689.     return 0;
  690. }
  691.  
  692.  
  693.  
  694. #else                /* ******** !WIN32 ******** */
  695.  
  696. /* Print File to port */
  697. private int
  698. gp_printfile(const char *filename, const char *pmport)
  699. {
  700. #define PRINT_BUF_SIZE 16384u
  701. char *buffer;
  702. char *portname;
  703. int i, port;
  704. FILE *f;
  705. DLGPROC lpfnSpoolProc;
  706. WORD count;
  707. DLGPROC lpfnCancelProc;
  708. int error = FALSE;
  709. long lsize;
  710. long ldone;
  711. char pcdone[20];
  712. MSG msg;
  713. HPJOB hJob;
  714.  
  715.     if (is_spool(pmport) && (strlen(pmport) >= 8)) {
  716.         /* translate from printer name to port name */
  717.         char driverbuf[256];
  718.         GetProfileString("Devices", pmport+8, "", driverbuf, sizeof(driverbuf));
  719.         strtok(driverbuf, ",");
  720.         pmport = strtok(NULL, ",");
  721.     }
  722.  
  723.     /* get list of ports */
  724.     if ((buffer = malloc(PRINT_BUF_SIZE)) == (char *)NULL)
  725.         return FALSE;
  726.  
  727.     if ( (strlen(pmport)==0) || (strcmp(pmport, "PRN")==0) ) {
  728.         GetProfileString("ports", NULL, "", buffer, PRINT_BUF_SIZE);
  729.         /* select a port */
  730. #ifdef __WIN32__
  731.         lpfnSpoolProc = (DLGPROC)SpoolDlgProc;
  732. #else
  733. #ifdef __DLL__
  734.         lpfnSpoolProc = (DLGPROC)GetProcAddress(phInstance, "SpoolDlgProc");
  735. #else
  736.         lpfnSpoolProc = (DLGPROC)MakeProcInstance((FARPROC)SpoolDlgProc, phInstance);
  737. #endif
  738. #endif
  739.         port = DialogBoxParam(phInstance, "SpoolDlgBox", (HWND)NULL, lpfnSpoolProc, (LPARAM)buffer);
  740. #if !defined(__WIN32__) && !defined(__DLL__)
  741.         FreeProcInstance((FARPROC)lpfnSpoolProc);
  742. #endif
  743.         if (!port) {
  744.             free(buffer);
  745.             return FALSE;
  746.         }
  747.         portname = buffer;
  748.         for (i=1; i<port && strlen(portname)!=0; i++)
  749.             portname += lstrlen(portname)+1;
  750.     }
  751.     else
  752.         portname = (char *)pmport;    /* Print Manager port name already supplied */
  753.     
  754.     if ((f = fopen(filename, "rb")) == (FILE *)NULL) {
  755.         free(buffer);
  756.         return FALSE;
  757.     }
  758.     fseek(f, 0L, SEEK_END);
  759.     lsize = ftell(f);
  760.     if (lsize <= 0)
  761.         lsize = 1;
  762.     fseek(f, 0L, SEEK_SET);
  763.  
  764.     hJob = OpenJob(portname, filename, (HPJOB)NULL);
  765.     switch ((int)hJob) {
  766.         case SP_APPABORT:
  767.         case SP_ERROR:
  768.         case SP_OUTOFDISK:
  769.         case SP_OUTOFMEMORY:
  770.         case SP_USERABORT:
  771.         fclose(f);
  772.         free(buffer);
  773.         return FALSE;
  774.     }
  775.     if (StartSpoolPage(hJob) < 0)
  776.         error = TRUE;
  777.  
  778. #ifdef __WIN32__
  779.     lpfnCancelProc = (DLGPROC)CancelDlgProc;
  780. #else
  781. #ifdef __DLL__
  782.     lpfnCancelProc = (DLGPROC)GetProcAddress(phInstance, "CancelDlgProc");
  783. #else
  784.     lpfnCancelProc = (DLGPROC)MakeProcInstance((FARPROC)CancelDlgProc, phInstance);
  785. #endif
  786. #endif
  787.     hDlgModeless = CreateDialog(phInstance, "CancelDlgBox", (HWND)NULL, lpfnCancelProc);
  788.     ldone = 0;
  789.  
  790.     while (!error && hDlgModeless 
  791.       && (count = fread(buffer, 1, PRINT_BUF_SIZE, f)) != 0 ) {
  792.         if (WriteSpool(hJob, buffer, count) < 0)
  793.         error = TRUE;
  794.         ldone += count;
  795.         sprintf(pcdone, "%d %%done", (int)(ldone * 100 / lsize));
  796.         SetWindowText(GetDlgItem(hDlgModeless, CANCEL_PCDONE), pcdone);
  797.         while (PeekMessage(&msg, hDlgModeless, 0, 0, PM_REMOVE)) {
  798.         if ((hDlgModeless == 0) || !IsDialogMessage(hDlgModeless, &msg)) {
  799.             TranslateMessage(&msg);
  800.             DispatchMessage(&msg);
  801.           }
  802.           }
  803.       }
  804.     free(buffer);
  805.     fclose(f);
  806.  
  807.     if (!hDlgModeless)
  808.         error=TRUE;
  809.     DestroyWindow(hDlgModeless);
  810.     hDlgModeless = 0;
  811. #if !defined(__WIN32__) && !defined(__DLL__)
  812.     FreeProcInstance((FARPROC)lpfnCancelProc);
  813. #endif
  814.     EndSpoolPage(hJob);
  815.     if (error)
  816.         DeleteJob(hJob, 0);
  817.     else
  818.         CloseJob(hJob);
  819.     return !error;
  820. }
  821.  
  822. #endif                /* ******** (!)WIN32 ******** */
  823.  
  824. /* ------ File naming and accessing ------ */
  825.  
  826. /* Create and open a scratch file with a given name prefix. */
  827. /* Write the actual file name at fname. */
  828. FILE *
  829. gp_open_scratch_file(const char *prefix, char *fname, const char *mode)
  830. {    char *temp;
  831.     if ( (temp = getenv("TEMP")) == NULL )
  832.         *fname = 0;
  833.     else
  834.     {    strcpy(fname, temp);
  835.         /* Prevent X's in path from being converted by mktemp. */
  836.         for ( temp = fname; *temp; temp++ )
  837.             *temp = tolower(*temp);
  838.         if ( strlen(fname) && (fname[strlen(fname)-1] != '\\') )
  839.             strcat(fname, "\\");
  840.     }
  841.     strcat(fname, prefix);
  842.     strcat(fname, "XXXXXX");
  843.     mktemp(fname);
  844.     return fopen(fname, mode);
  845. }
  846.  
  847. /* Open a file with the given name, as a stream of uninterpreted bytes. */
  848. FILE *
  849. gp_fopen(const char *fname, const char *mode)
  850. {    return fopen(fname, mode);
  851. }
  852.  
  853.